home *** CD-ROM | disk | FTP | other *** search
- #!/usr/bin/perl
- # Debian task selector, mark II.
- # Copyright 2004-2006 by Joey Hess <joeyh@debian.org>.
- # Licensed under the GPL, version 2 or higher.
- use Locale::gettext;
- use Getopt::Long;
- use POSIX qw(floor);
- use warnings;
- use strict;
- textdomain('tasksel');
-
- my $debconf_helper="/usr/lib/tasksel/tasksel-debconf";
- my $filter_tasks="/usr/lib/tasksel/filter-tasks";
- my $disconnect="/usr/lib/tasksel/disconnect";
- my $testdir="/usr/lib/tasksel/tests";
- my $packagesdir="/usr/lib/tasksel/packages";
- my $descdir="/usr/share/tasksel";
- my $localdescdir="/usr/local/share/tasksel";
- my $statusfile="/var/lib/dpkg/status";
- my $infodir="/usr/lib/tasksel/info";
- my $testmode=0;
-
- sub warning {
- print STDERR "tasksel: @_\n";
- }
-
- sub error {
- print STDERR "tasksel: @_\n";
- exit 1;
- }
-
- # Run a shell command except in test mode, and returns its exit code.
- # Prints the command in test mode. Parameters should be pre-split for
- # system.
- sub run {
- if ($testmode) {
- print join(" ", @_)."\n";
- return 0;
- }
- else {
- return system(@_) >> 8;
- }
- }
-
- sub run_disconnect {
- if ($testmode) {
- print join(" ", @_)."\n";
- return 0;
- }
- else {
- return system($disconnect, @_) >> 8;
- }
- }
-
- # A list of all available task desc files.
- sub list_task_descs {
- return glob("$descdir/*.desc"), glob("$localdescdir/*.desc");
- }
-
- # Returns a list of hashes; hash values are arrays for multi-line fields.
- sub read_task_desc {
- my $desc=shift;
- my @ret;
- open (DESC, "<$desc") || die "read $desc\: $!";
- local $/="\n\n";
- while (<DESC>) {
- my %data;
- my @lines=split("\n");
- while (@lines) {
- my $line=shift(@lines);
- if ($line=~/^([^ ]+):(?: (.*))?/) {
- my ($key, $value)=($1, $2);
- $key=lc($key);
- if (@lines && $lines[0] =~ /^\s+/) {
- # multi-line field
- my @values;
- if (defined $value && length $value) {
- push @values, $value;
- }
- while (@lines && $lines[0] =~ /^\s+(.*)/) {
- push @values, $1;
- shift @lines;
- }
- $data{$key}=[@values];
- }
- else {
- $data{$key}=$value;
- }
- }
- else {
- warning "parse error in stanza $. of $desc";
- }
- }
- if (%data) {
- $data{relevance}=5 unless exists $data{relevance};
- $data{shortdesc}=$data{description}->[0];
- $data{shortdesctrans}=dgettext("debian-tasks", $data{shortdesc});
- push @ret, \%data;
- }
- }
- close DESC;
- return @ret;
- }
-
- # Loads info for all tasks, and returns a set of task structures.
- sub all_tasks {
- my %seen;
- grep { $seen{$_->{task}}++; $seen{$_->{task}} < 2 }
- map { read_task_desc($_) } list_task_descs();
- }
-
- # Returns a list of all available packages.
- sub list_avail {
- my @list;
- # Might be better to use the perl apt bindings, but they are not
- # currently in base.
- open (AVAIL, "apt-cache dumpavail|");
- local $_;
- while (<AVAIL>) {
- chomp;
- if (/^Package: (.*)/) {
- push @list, $1;
- }
- }
- close AVAIL;
- return @list;
- }
-
- # Returns a list of all installed packages.
- sub list_installed {
- my @list;
- local $/="\n\n";
- open (STATUS, $statusfile);
- local $_;
- while (<STATUS>) {
- if (/^Status: .* installed$/m && /Package: (.*)$/m) {
- push @list, $1;
- }
- }
- close STATUS;
- return @list;
- }
-
- my %avail_pkgs;
- # Given a package name, checks to see if it's available. Memoised.
- sub package_avail {
- my $package=shift;
-
- if (! %avail_pkgs) {
- foreach my $pkg (list_avail()) {
- $avail_pkgs{$pkg} = 1;
- }
- }
-
- return $avail_pkgs{$package} || package_installed($package);
- }
-
- my %installed_pkgs;
- # Given a package name, checks to see if it's installed. Memoised.
- sub package_installed {
- my $package=shift;
-
- if (! %installed_pkgs) {
- foreach my $pkg (list_installed()) {
- $installed_pkgs{$pkg} = 1;
- }
- }
-
- return $installed_pkgs{$package};
- }
-
- # Given a task hash, checks if its key packages are available.
- sub task_avail {
- local $_;
- my $task=shift;
- if (! ref $task->{key}) {
- return 1;
- }
- else {
- foreach my $pkg (@{$task->{key}}) {
- if (! package_avail($pkg)) {
- return 0;
- }
- }
- return 1;
- }
- }
-
- # Given a task hash, checks to see if it is already installed.
- # (All of its key packages must be installed.)
- sub task_installed {
- local $_;
- my $task=shift;
- if (! ref $task->{key}) {
- return 0; # can't tell with no key packages
- }
- else {
- foreach my $pkg (@{$task->{key}}) {
- if (! package_installed($pkg)) {
- return 0;
- }
- }
- return 1;
- }
- }
-
- # Given task hash, returns a list of all available packages in the task.
- # If the aptitude_tasks or apt_get_tasks parameter is true, then it does
- # not expand tasks that aptitude/apt-get knows about, and just returns
- # aptitude/apt-get task syntax for those.
- sub task_packages {
- my $task=shift;
- my $aptitude_tasks=shift;
- my $apt_get_tasks=shift;
-
- my %list;
-
- # key packages are always included
- if (ref $task->{key}) {
- map { $list{$_}=1 } @{$task->{key}};
- }
-
- if ($task->{packages} eq 'task-fields') {
- # task-fields method is built-in for speed and to support
- # aptitude/apt-get task definitions
- if ($aptitude_tasks) {
- return '~t^'.$task->{task}.'$';
- }
- elsif ($apt_get_tasks) {
- return $task->{task}.'^';
- }
- else {
- local $/="\n\n";
- open (AVAIL, "apt-cache dumpavail|");
- while (<AVAIL>) {
- if (/^Task: (.*)/m) {
- my @tasks=split(", ", $1);
- if (grep { $_ eq $task->{task} } @tasks) {
- $list{$1}=1 if /^Package: (.*)/m;
- }
- }
- }
- close AVAIL;
- }
- }
- elsif ($task->{packages} eq 'standard') {
- # standard method is built in since it cannot easily be
- # implemented externally.
- #return "~pstandard", "~prequired", "~pimportant";
- # Unfortunately, this needs aptitude, not apt-get, and we
- # have to use apt-get for other reasons ...
- die "'Packages: standard' not supported by this version of tasksel\n";
- }
- elsif ($task->{packages} eq 'manual') {
- # manual package selection is a special case
- return;
- }
- else {
- # external method
- my ($method, @params);
- if (ref $task->{packages}) {
- @params=@{$task->{packages}};
- $method=shift @params;
- }
- else {
- $method=$task->{packages};
- }
-
- map { $list{$_}=1 }
- grep { package_avail($_) }
- split(' ', `$packagesdir/$method $task->{task} @params`);
- }
-
- return keys %list;
- }
-
- # Calculate the list of packages to remove, taking care not to include
- # packages that are in other tasks that are still selected.
- sub task_packages_remove {
- my $tasks_remove = shift;
-
- my %list;
-
- foreach my $task (@$tasks_remove) {
- $list{$_} = 1 foreach task_packages($task, 0, 0);
- }
- return keys %list unless %list;
- foreach my $task (grep { ! $_->{_display} || $_->{_selected} } @_) {
- delete $list{$_} foreach task_packages($task, 0, 0);
- }
-
- return keys %list;
- }
-
- # Given a list of task hashes, adjust their statuses according to
- # tasksel/force-tasks, tasksel/limit-tasks, and tasksel/skip-tasks.
- sub filter_tasks {
- my %tasks=map { $_->{task} => $_ } @_;
- my $tmpfile=`tempfile`;
- chomp $tmpfile;
- my $ret=system($filter_tasks, $tmpfile, sort keys %tasks) >> 8;
- if ($ret != 0) {
- error "filter-tasks failed to run";
- }
- open(IN, "<$tmpfile") or return;
- for my $line (<IN>) {
- chomp $line;
- $line=~/(.*?) (install|skip)/ or next;
- my $name=$1;
- my $action=$2;
- if ($action eq 'install') {
- $tasks{$name}{_display} = 0;
- $tasks{$name}{_install} = 1;
- }
- elsif ($action eq 'skip') {
- $tasks{$name}{_display} = 0;
- $tasks{$name}{_install} = 0;
- }
- }
- close IN;
- unlink $tmpfile;
- }
-
- # Given a task hash, runs any test program specified in its data, and sets
- # the _display and _install fields to 1 or 0 depending on its result.
- sub task_test {
- my $task=shift;
- my $new_install=shift;
- $task->{_display} = shift; # default
- $task->{_install} = shift; # default
- $ENV{NEW_INSTALL}=$new_install if defined $new_install;
- foreach my $test (grep /^test-.*/, keys %$task) {
- $test=~s/^test-//;
- if (-x "$testdir/$test") {
- my $ret=system("$testdir/$test", $task->{task}, split " ", $task->{"test-$test"}) >> 8;
- if ($ret == 0) {
- $task->{_display} = 0;
- $task->{_install} = 1;
- }
- elsif ($ret == 1) {
- $task->{_display} = 0;
- $task->{_install} = 0;
- }
- elsif ($ret == 2) {
- $task->{_display} = 1;
- $task->{_install} = 1;
- }
- elsif ($ret == 3) {
- $task->{_display} = 1;
- $task->{_install} = 0;
- }
- }
- }
-
- delete $ENV{NEW_INSTALL};
- return $task;
- }
-
- # Hides a task if it enhances other tasks. Don't mark it not to be installed
- # here; this will be taken care of later outside new-install mode, and in
- # new-install mode we may want to apply a test to an enhancing task that
- # installs it by default.
- sub hide_enhancing_tasks {
- my $task=shift;
- if (exists $task->{enhances} && length $task->{enhances}) {
- $task->{_display} = 0;
- }
- return $task;
- }
-
- # Converts a list of tasks into a debconf list of their short descriptions.
- sub task_to_debconf {
- my $field = shift;
- join ", ", map {
- my $desc=$_->{$field};
- if ($desc=~/, /) {
- warning("task ".$_->{task}." contains a comma in its short description: \"$desc\"");
- }
- $desc;
- } @_;
- }
-
- # Given a first parameter that is a debconf list of short descriptions of
- # tasks, or a dependency style list of task names, and then a list of task
- # hashes, returns a list of hashes for all the tasks in the list.
- sub list_to_tasks {
- my $list=shift;
- my %desc_to_task = map { $_->{shortdesc} => $_, $_->{task} => $_ } @_;
- return grep { defined } map { $desc_to_task{$_} } split ", ", $list;
- }
-
- # Orders a list of tasks for display.
- sub order_for_display {
- sort {
- $b->{relevance} <=> $a->{relevance}
- || 0 ||
- $a->{section} cmp $b->{section}
- || 0 ||
- $a->{shortdesc} cmp $b->{shortdesc}
- } @_;
- }
-
- # Given a set of tasks and a name, returns the one with that name.
- sub name_to_task {
- my $name=shift;
- return (grep { $_->{task} eq $name } @_)[0];
- }
-
- sub find_task_script {
- my $task=shift;
- my $script=shift;
-
- my $path="$infodir/$task.$script";
- if (-e $path && -x _) {
- return $path;
- } else {
- return undef;
- }
- }
-
- sub run_task_script {
- my $path=pop;
- my @prefix=@_;
-
- my $ret=run(@prefix, $path);
- if ($ret != 0) {
- warning("$path exited with nonzero code $ret");
- return 0;
- }
- return 1;
- }
-
- sub debconf_apt_command {
- my $dap=shift;
- my $from=shift;
- my $to=shift;
- my $options=shift;
-
- my @cmd=($dap, '--from', $from, '--to', $to);
- push @cmd, @$options if defined $options;
- push @cmd, '--';
- return @cmd;
- }
-
- # Works out the region of a progress bar needed for a given invocation of
- # debconf-apt-progress, and returns the appropriate debconf-apt-progress
- # command.
- sub task_region {
- my $dap=shift;
- my $start=shift;
- my $end=shift;
- my $options=shift;
- my $cur_region=shift;
- my $step_region=shift;
- my $num_regions=shift;
-
- if (defined $start and defined $end and $num_regions) {
- my $from=floor($start + ($end - $start) * $cur_region / $num_regions + 0.5);
- my $to=floor($start + ($end - $start) * ($cur_region + $step_region) / $num_regions + 0.5);
- return debconf_apt_command($dap, $from, $to, $options);
- } else {
- return;
- }
- }
-
- sub usage {
- print STDERR gettext(q{Usage:
- tasksel install <task>
- tasksel remove <task>
- tasksel [options]
- -t, --test test mode; don't really do anything
- --new-install automatically install some tasks
- --list-tasks list tasks that would be displayed and exit
- --task-packages list available packages in a task
- --task-desc returns the description of a task
- });
- }
-
- # Process command line options and return them in a hash.
- sub getopts {
- my %ret;
- Getopt::Long::Configure ("bundling");
- if (! GetOptions(\%ret, "test|t", "new-install", "list-tasks",
- "task-packages=s@", "task-desc=s",
- "section=s",
- "debconf-apt-from=i", "debconf-apt-to=i",
- "debconf-apt-progress=s")) {
- usage();
- exit(1);
- }
- # Special case apt-like syntax.
- if (@ARGV && $ARGV[0] eq "install") {
- shift @ARGV;
- $ret{install} = shift @ARGV;
- }
- if (@ARGV && $ARGV[0] eq "remove") {
- shift @ARGV;
- $ret{remove} = shift @ARGV;
- }
- if (@ARGV) {
- usage();
- exit 1;
- }
- $testmode=1 if $ret{test}; # set global
- return %ret;
- }
-
- sub main {
- my %options=getopts();
- my @tasks_remove;
- my @tasks_install;
-
- # Options that output stuff and don't need a full processed list of
- # tasks.
- if (exists $options{"task-packages"}) {
- my @tasks=all_tasks();
- foreach my $taskname (@{$options{"task-packages"}}) {
- my $task=name_to_task($taskname, @tasks);
- if ($task) {
- print "$_\n" foreach task_packages($task);
- }
- }
- exit(0);
- }
- elsif ($options{"task-desc"}) {
- my $task=name_to_task($options{"task-desc"}, all_tasks());
- if ($task) {
- my $extdesc=join(" ", @{$task->{description}}[1..$#{$task->{description}}]);
- print dgettext("debian-tasks", $extdesc)."\n";
- exit(0);
- }
- else {
- exit(1);
- }
- }
-
- # This is relatively expensive, get the full list of available tasks and
- # mark them.
- my @tasks=map { hide_enhancing_tasks($_) } map { task_test($_, $options{"new-install"}, 1, 0) }
- grep { task_avail($_) } all_tasks();
- if ($options{"new-install"}) {
- filter_tasks(@tasks);
- }
-
- # Limit the tasks shown to a single section
- if ($options{section}) {
- @tasks = grep { $options{section} eq $_->{section} } @tasks;
- }
-
- if ($options{"list-tasks"}) {
- map { $_->{_installed} = task_installed($_) } @tasks;
- print "".($_->{_installed} ? "i" : "u")." ".$_->{task}."\t".$_->{shortdesc}."\n"
- foreach order_for_display(grep { $_->{_display} } @tasks);
- exit(0);
- }
-
- if (! $options{"new-install"}) {
- # Don't install hidden tasks if this is not a new install.
- map { $_->{_install} = 0 } grep { $_->{_display} == 0 } @tasks;
- }
- if ($options{"install"}) {
- my $task=name_to_task($options{"install"}, @tasks);
- $task->{_install} = 1 if $task;
- }
- if ($options{"remove"}) {
- my $task=name_to_task($options{"remove"}, @tasks);
- push @tasks_remove, $task;
- }
-
- # The interactive bit.
- my $interactive=0;
- my @list = order_for_display(grep { $_->{_display} == 1 } @tasks);
- if (@list && ! $options{install} && ! $options{remove}) {
- $interactive=1;
- if (! $options{"new-install"}) {
- # Find tasks that are already installed.
- map { $_->{_installed} = task_installed($_) } @list;
- # Don't install new tasks unless manually selected.
- map { $_->{_install} = 0 } @list;
- }
- else {
- # Assume that no tasks are installed, to ensure
- # that complete tasks get installed on new
- # installs.
- map { $_->{_installed} = 0 } @list;
- }
- my $question="tasksel/tasks";
- if ($options{"new-install"}) {
- $question="tasksel/first";
- }
- my @default = grep { $_->{_display} == 1 && ($_->{_install} == 1 || $_->{_installed} == 1) } @tasks;
- my $tmpfile=`tempfile`;
- chomp $tmpfile;
-
- # Canonicalise $question's value first, so that it will be
- # displayed properly if preseeded unseen.
- my $ret=system($debconf_helper, "--get", $tmpfile,
- $question) >> 8;
- if ($ret != 0) {
- error "debconf failed to run";
- }
- open(IN, "<$tmpfile");
- $ret=<IN>;
- if (! defined $ret) {
- die "tasksel canceled\n";
- }
- chomp $ret;
- close IN;
-
- if ($ret ne "") {
- my $new_value = task_to_debconf("shortdesc",
- list_to_tasks($ret, @list));
- $ret=system($debconf_helper, "--set", $question,
- $new_value) >> 8;
- if ($ret != 0) {
- error "debconf failed to run";
- }
- }
-
- $ret=system($debconf_helper, $tmpfile,
- task_to_debconf("shortdesc", @list),
- task_to_debconf("shortdesctrans", @list),
- task_to_debconf("shortdesc", @default),
- $question) >> 8;
- if ($ret == 30) {
- exit 10; # back up
- }
- elsif ($ret != 0) {
- error "debconf failed to run";
- }
- open(IN, "<$tmpfile");
- $ret=<IN>;
- if (! defined $ret) {
- die "tasksel canceled\n";
- }
- chomp $ret;
- close IN;
- unlink $tmpfile;
-
- # Set _install flags based on user selection.
- map { $_->{_install} = 0 } @list;
- foreach my $task (list_to_tasks($ret, @tasks)) {
- if (! $task->{_installed}) {
- $task->{_install} = 1;
- }
- $task->{_selected} = 1;
- }
- foreach my $task (@list) {
- if (! $task->{_selected} && $task->{_installed}) {
- push @tasks_remove, $task;
- }
- }
- }
-
- # If an enhancing task is already marked for
- # install, probably by preseeding, mark the tasks
- # it enhances for install.
- foreach my $task (grep { $_->{_install} && exists $_->{enhances} &&
- length $_->{enhances} } @tasks) {
- # Ignore any disjunctive sets. We don't know which
- # element(s) we should use, so if the user didn't select any
- # it's better to just leave them alone.
- my $simple_enhances=join ", ", grep { ! /\|/ } split ", ", $task->{enhances};
- map { $_->{_install}=1 } list_to_tasks($simple_enhances, @tasks);
- }
-
- # Select enhancing tasks for install.
- # XXX FIXME ugly hack -- loop until enhances settle to handle
- # chained enhances. This is sloow and could loop forever if
- # there's a cycle.
- my $enhances_needswork=1;
- while ($enhances_needswork) {
- $enhances_needswork=0;
- foreach my $task (grep { ! $_->{_install} && exists $_->{enhances} &&
- length $_->{enhances} } @tasks) {
- my @ordeps=split ", ", $task->{enhances};
- if (@ordeps) {
- my $orig_state=$task->{_install};
-
- # Mark enhancing tasks for install if their
- # dependencies are met and their test fields
- # mark them for install.
- $ENV{TESTING_ENHANCER}=1;
- task_test($task, $options{"new-install"}, 0, 1);
- delete $ENV{TESTING_ENHANCER};
- foreach my $ordep (@ordeps) {
- my $satisfied=0;
- foreach my $dep (list_to_tasks(join(", ", split " | ", $ordep), @tasks)) {
- if ($dep->{_install}) {
- $satisfied=1;
- }
- }
- if (! $satisfied) {
- $task->{_install} = 0;
- }
- }
-
- if ($task->{_install} != $orig_state) {
- $enhances_needswork=1;
- }
- }
- }
- }
-
- # Add tasks to install and see if any selected task requires manual
- # selection.
- my $manual_selection=0;
- foreach my $task (grep { $_->{_install} } @tasks) {
- push @tasks_install, $task;
- if ($task->{packages} eq 'manual') {
- if (-x "/usr/bin/aptitude") {
- $manual_selection=1;
- } else {
- warning "task $task requires aptitude";
- }
- }
- }
-
- my $dap;
- my ($dap_from, $dap_to);
- my @dap_options;
- my (@aptitude, @apt);
- if (-x "/usr/bin/debconf-apt-progress") {
- $dap="debconf-apt-progress";
- if (exists $options{'debconf-apt-from'} and
- exists $options{'debconf-apt-to'}) {
- $dap_from=$options{'debconf-apt-from'};
- $dap_to=$options{'debconf-apt-to'};
- }
- push @dap_options, split(' ', $options{'debconf-apt-progress'})
- if exists $options{'debconf-apt-progress'};
- @aptitude=qw{aptitude -q};
- @apt=qw{apt-get -q};
- }
- else {
- @aptitude="aptitude";
- @apt="apt-get";
- }
-
- # We arbitrarily allocate 5% of the progress bar to task scripts,
- # divided equally among all the scripts. (If you change this, the
- # new percentage should divide equally into 100.)
- my $num_regions=0;
- foreach my $task (@tasks_remove) {
- if (defined find_task_script($task->{task}, "prerm")) {
- ++$num_regions;
- }
- if (defined find_task_script($task->{task}, "postrm")) {
- ++$num_regions;
- }
- }
- foreach my $task (@tasks_install) {
- if (defined find_task_script($task->{task}, "preinst")) {
- ++$num_regions;
- }
- if (defined find_task_script($task->{task}, "postinst")) {
- ++$num_regions;
- }
- }
- $num_regions*=20;
- my $cur_region=0;
-
- my @task_packages_remove = task_packages_remove(\@tasks_remove, @tasks);
-
- # And finally, act on selected tasks.
- if (@tasks_install || @task_packages_remove || $manual_selection) {
- my (@aptitude_args, @apt_args);
- my @args = map { "$_-" } @task_packages_remove;
- push @aptitude_args, @args;
- push @apt_args, @args;
- foreach my $task (@tasks_remove) {
- my $path=find_task_script($task->{task}, "prerm");
- if (defined $path) {
- my @prefix=task_region($dap, $dap_from, $dap_to, \@dap_options, $cur_region++, 1, $num_regions);
- run_task_script(@prefix, $path);
- }
- }
- foreach my $task (@tasks_install) {
- push @aptitude_args, task_packages($task, 1, 0);
- push @apt_args, task_packages($task, 0, 1);
- my $path=find_task_script($task->{task}, "preinst");
- if (defined $path) {
- my @prefix=task_region($dap, $dap_from, $dap_to, \@dap_options, $cur_region++, 1, $num_regions);
- run_task_script(@prefix, $path);
- }
- }
- my @prefix;
- if (defined $dap_from and defined $dap_to) {
- if ($num_regions) {
- # The main apt-get run takes up the
- # remaining 95% of the progress bar.
- @prefix=task_region($dap, $dap_from, $dap_to, \@dap_options, $cur_region, $num_regions * 0.95, $num_regions);
- $cur_region=floor($cur_region + $num_regions * 0.95 + 0.5);
- } else {
- @prefix=debconf_apt_command($dap, $dap_from, $dap_to, \@dap_options);
- }
- } elsif (defined $dap) {
- @prefix=($dap, @dap_options, '--');
- }
- my $ret;
- # If the user selected no other tasks and manual package
- # selection, run aptitude w/o the --visual-preview parameter.
- if (! @aptitude_args && $manual_selection) {
- $ret=run_disconnect("aptitude", "--schedule-only");
- if ($ret != 0) {
- error gettext("aptitude failed")." ($ret)";
- } else {
- $ret=run(@prefix, @aptitude, "-y", "install");
- if ($ret != 0) {
- error gettext("aptitude failed")." ($ret)";
- }
- }
- }
- else {
- if ($manual_selection) {
- $ret=run(@aptitude, "-y", "--schedule-only", "install", @aptitude_args);
- } else {
- $ret=run(@prefix, @apt, "-y", "install", @apt_args);
- }
- if ($ret != 0) {
- error gettext("aptitude failed")." ($ret)";
- } elsif ($manual_selection) {
- $ret=run_disconnect(@aptitude, "--schedule-only");
- if ($ret != 0) {
- error gettext("aptitude failed")." ($ret)";
- } else {
- $ret=run(@prefix, @aptitude, "-y", "install");
- if ($ret != 0) {
- error gettext("aptitude failed")." ($ret)";
- }
- }
- }
- }
- foreach my $task (@tasks_remove) {
- my $path=find_task_script($task->{task}, "postrm");
- if (defined $path) {
- my @prefix=task_region($dap, $dap_from, $dap_to, \@dap_options, $cur_region++, 1, $num_regions);
- run_task_script(@prefix, $path);
- }
- }
- foreach my $task (@tasks_install) {
- my $path=find_task_script($task->{task}, "postinst");
- if (defined $path) {
- my @prefix=task_region($dap, $dap_from, $dap_to, \@dap_options, $cur_region++, 1, $num_regions);
- run_task_script(@prefix, $path);
- }
- }
- }
- }
-
- main();
-